home *** CD-ROM | disk | FTP | other *** search
Text File | 1997-05-28 | 28.8 KB | 1,193 lines |
- import "constant.ai";
-
- ' Forward declarations
- procedure AttackInZOC(movingUnit);
- procedure ShootMissiles(shootingUnit);
- procedure Defend(hexToDefend);
- procedure RecoverUnits();
- procedure RecoverAnyUnit();
- procedure Rally();
- procedure PartialRecover();
- procedure GetUnitsInRange(list);
- procedure GetUnitsOutOfRange(list);
- procedure MoveLeader();
- procedure EnemyWithinRangeOfGroup(radius);
- procedure ShootMissilesFar(unitToFire);
- procedure CommandOtherUnits();
- procedure NoMoreUnitsToCommand();
- procedure TurnToDefend();
- procedure GetUnitsToCommand(legalList);
-
- ' GetFrontalFacings takes a list id and a facing (direction). It fills the
- ' list with the (2) directions of that a unit can go. It returns 1 if the
- ' original facing is correct, otherwise it returns 0. (NORTH and SOUTH are
- ' NOT valid facings).
- procedure GetFrontalFacings(listID, facing)
- {
- var retVal;
-
- retVal = 0;
- if (facing == NORTHEAST)
- {
- AddToList(listID, NORTH);
- AddToList(listID, NORTHEAST);
- retVal = 1;
- };
- if (facing == EAST)
- {
- AddToList(listID, NORTHEAST);
- AddToList(listID, SOUTHEAST);
- retVal = 1;
- };
- if (facing == SOUTHEAST)
- {
- AddToList(listID, SOUTHEAST);
- AddToList(listID, SOUTH);
- retVal = 1;
- };
- if (facing == SOUTHWEST)
- {
- AddToList(listID, SOUTH);
- AddToList(listID, SOUTHWEST);
- retVal = 1;
- };
- if (facing == WEST)
- {
- AddToList(listID, SOUTHWEST);
- AddToList(listID, NORTHWEST);
- retVal = 1;
- };
- if (facing == NORTHWEST)
- {
- AddToList(listID, NORTHWEST);
- AddToList(listID, NORTH);
- retVal = 1;
- };
- return retVal;
- };
-
- ' inProximity looks at the identifying hex lists (flank, rear, or frontal)
- ' to see if the hex in question is in that list. It returns TRUE if it is,
- ' otherwise it returns FALSE. For the identifying list types it is:
- ' 0 = ZOC
- ' 1 = flank
- ' 2 = rear
- procedure inProximity(hexInQuestion, nation, hexType)
- {
- var dir, hex, hexList, retVal, unit, n, isLeader;
-
- retVal = FALSE;
-
- for dir = NORTH to NORTHWEST
- {
- hex = AdjacentHex(hexInQuestion, dir);
- if (hex != -1)
- {
- unit = HexOccupancy(hex);
- if (unit!=-1)
- {
- isLeader = IsLeader(unit);
- if (NOT isLeader)
- {
- n = Query(NATION, unit);
- if (n != nation)
- {
- hexList = GetList();
- AssignList(hexList, hexType, unit); ' Assign flank hexes
- if (IsMemberOf(hexList, hexInQuestion))
- retVal = TRUE;
- };
- ' ReleaseList(hexList);
- };
- };
- };
- };
- return retVal;
- };
-
- '**************************************************************************
- ' InFlank checks to see if the hex in question is in the flank of an enemy
- ' unit.
- procedure InFlank(hexInQuestion, nation)
- {
- return inProximity(hexInQuestion, nation, 1); ' 1 identifies flank hexes
- };
-
- '**************************************************************************
- ' InRear checks to see if the hex in question is in the rear of an enemy
- ' unit.
- procedure InRear(hexInQuestion, nation)
- {
- return inProximity(hexInQuestion, nation, 2); ' 2 identifies rear hexes
- };
-
- '**************************************************************************
- ' InFrontal checks to see if the hex in question is in the frontal of an
- ' enemy unit.
- procedure InFrontal(hexInQuestion, nation)
- {
- return inProximity(hexInQuestion, nation, 0); ' 0 identifies frontal hexes
- };
-
- '**************************************************************************
- ' WeightCombatPath looks at the path of a combat unit and weights it
- ' according to the following:
- ' Clear = 9
- ' Rear = 7
- ' Flank = 5
- ' Frontal = 2
- ' DS = 0
- ' These values may be modified in the CONSTANT.AI file.
- ' Currently (7/31/96), only the final hex in the path is looked at. That
- ' may change to be every hex is looked at.
- procedure WeightCombatPath(pathID, nation)
- {
- var hexInQuestion, weight, w;
-
- ' Assume the path is going to be clear.
- weight = CLEAR;
-
- hexInQuestion = GetLast(pathID);
- w = InFlank(hexInQuestion, nation);
- if (w == TRUE)
- {
- weight = FLANK;
- };
- ' A rear attack would be much more desirable than a flank attack so that
- ' why it is checked for after the flank
- if (InRear(hexInQuestion, nation) == TRUE)
- {
- weight = REAR;
- };
- ' A frontal attack is the least desirable but it needs to be checked for
- ' last since it is something that the unit would not want to move into
- if (InFrontal(hexInQuestion, nation) == TRUE)
- {
- weight = FRONTAL;
- };
-
- return weight;
- };
-
- procedure GenericMove(hex);
-
- procedure PersianFlankingPosAcheived(hex)
- {
- var unit, retVal, n;
-
- unit = HexOccupancy(hex);
- retVal = False;
-
- if (unit != -1)
- {
- n = Query(NATION, unit);
- if (n == PERSIAN)
- retVal = True;
- };
- return retVal;
- };
-
- procedure IndianFlankingPosAcheived(hex)
- {
- var unit, retVal, n;
-
- unit = HexOccupancy(hex);
- retVal = False;
-
- if (unit != -1)
- {
- n = Query(NATION, unit);
- if (n == INDIAN)
- retVal = True;
- };
- return retVal;
- };
-
- procedure DanubianFlankingPosAcheived(hex)
- {
- var unit, retVal, n;
-
- unit = HexOccupancy(hex);
- retVal = False;
-
- if (unit != -1)
- {
- n = Query(NATION, unit);
- if (n == DANUBIAN)
- retVal = True;
- };
- return retVal;
- };
-
- procedure MacedonianFlankingPosAcheived(hex)
- {
- var unit, retVal, n;
-
- unit = HexOccupancy(hex);
- retVal = False;
-
- if (unit != -1)
- {
- n = Query(NATION, unit);
- if (n == MACEDONIAN)
- retVal = True;
- };
- return retVal;
- };
-
- procedure GreekFlankingPosAcheived(hex)
- {
- var unit, retVal, n;
-
- unit = HexOccupancy(hex);
- retVal = False;
-
- if (unit != -1)
- {
- n = Query(NATION, unit);
- if (n == GREEK)
- retVal = True;
- };
- return retVal;
- };
-
- procedure ScythianFlankingPosAcheived(hex)
- {
- var unit, retVal, n;
-
- unit = HexOccupancy(hex);
- retVal = False;
-
- if (unit != -1)
- {
- n = Query(NATION, unit);
- if (n == SCYTHIAN)
- retVal = True;
- };
- return retVal;
- };
-
- procedure DoNothing()
- {
- };
-
- procedure GetUnitsOutOfRange(list)
- {
- var unit, numOutOfRange;
-
- GroupListReset();
- numOutOfRange = 0;
- unit = GroupListGetNext();
- while (unit != -1)
- {
- if (NOT InCommandRange(unit))
- {
- AddToList(list, unit);
- numOutOfRange = numOutOfRange+1;
- };
- unit = GroupListGetNext();
- };
- return numOutOfRange;
- };
-
- procedure GetUnitsInRange(list)
- {
- var unit, numInRange;
-
- GroupListReset();
- numInRange = 0;
- unit = GroupListGetNext();
- while (unit != -1)
- {
- if (InCommandRange(unit))
- {
- AddToList(list, unit);
- numInRange = numInRange+1;
- };
- unit = GroupListGetNext();
- };
- return numInRange;
- };
-
- ' This procedure will tell how many of the units are engaged in battle.
- ' The definition of "engaged in battle" means if someone is in his ZOC OR
- ' he is in someone else's ZOC.
- procedure GetUnitsEngagedInBattle(list)
- {
- var unit, nation, numEngagedInBattle, surroundingHexes, zocHexes, hex,
- enemy, enemyNation, added, unitHex, leader, isLeader;
-
- GroupListReset();
- numEngagedInBattle = 0;
- leader = GetLeader();
- watch "Leader = " leader;
- unit = GroupListGetNext();
- if (unit == leader)
- unit = GroupListGetNext();
- while (unit != -1)
- {
- nation = Query(NATION, unit);
- added = False;
- unitHex = Query(REGION_NUM, unit);
- ' First, check to see if he has someone in his ZOC.
- zocHexes = GetList();
- AssignList(zocHexes, 0, unit); ' The 0 stands for ZOC hexes.
- ResetList(zocHexes);
- hex = GetNextInList(zocHexes);
- while ((hex!=-1) AND (NOT added))
- {
- enemy = HexOccupancy(hex);
- if (enemy != -1)
- {
- enemyNation = Query(NATION, enemy);
- if (enemyNation != nation)
- {
- ' Found an enemy in his ZOC... add him to the list.
- AddToList(list, unit);
- added = True;
- numEngagedInBattle = numEngagedInBattle + 1;
- };
- };
- hex = GetNextInList(zocHexes);
- };
- ' ReleaseList(zocHexes);
- ' Now check to see if anyone has him in his ZOC.
- surroundingHexes = GetList();
- AssignList(surroundingHexes, 5, unit); ' The 5 stands for surrounding hexes.
- ResetList(surroundingHexes);
- hex = GetNextInList(surroundingHexes);
- while ((hex != -1) AND (NOT added))
- {
- enemy = HexOccupancy(hex);
- if (enemy != -1)
- {
- enemyNation = Query(NATION, enemy);
- if (enemyNation != nation)
- {
- isLeader = IsLeader(enemy);
- if (NOT isLeader)
- {
- ClearList(zocHexes);
- AssignList(zocHexes, 0, enemy); ' The 0 stands for ZOC hexes
- ResetList(zocHexes);
- if (IsMemberOf(zocHexes, unitHex))
- {
- AddToList(list, unit);
- added = True;
- numEngagedInBattle = numEngagedInBattle + 1;
- };
- };
- };
- };
- hex = GetNextInList(surroundingHexes);
- };
- ' ReleaseList(surroundingHexes);
- unit = GroupListGetNext();
- if (unit == leader)
- unit = GroupListGetNext();
- };
- return numEngagedInBattle;
- };
-
- procedure ParePathListDown(pathList, destHex)
- {
- var path, hex, shortestDist, d, tempPath, tPath;
-
- d = SizeOfList(pathList);
- ResetList(pathList);
- shortestDist = 100;
- path = GetNextInList(pathList);
- while (path!=-1)
- {
- hex = GetLast(path);
- d = Distance(hex, destHex);
- if (d<shortestDist)
- {
- shortestDist = d;
- ' Remove all lists up to this point.
- ResetList(pathList);
- tempPath = GetNextInList(pathList);
- while (tempPath != path)
- {
- tPath = tempPath;
- tempPath = GetNextInList(pathList);
- RemoveFromList(pathList, tPath);
- ' ReleaseList(tPath);
- };
- ResetList(pathList);
- tPath = -1;
- }
- else
- {
- if (d<shortestDist)
- tPath = path;
- };
- path = GetNextInList(pathList);
- if (tPath != -1)
- {
- ' RemoveFromList(pathList, tPath);
- ' ReleaseList(tPath);
- };
- };
- d = SizeOfList(pathList);
- watch "Size of pathList: " d;
- };
-
- ' This procedure looks at the current units that can move, weights their
- ' paths and choose which unit to move. The units that are unable to move
- ' need to be passed into the unitsUnableToMove. This functions should only
- ' be called when the conditions are clear (no unit in the group is in
- ' battle). If a unit cannot move because his frontal hexes are covered (by
- ' friendlies), he is also not considered.
- ' If a unit is forced into a non-clear path, the most advantageous unit is
- ' moved to the most advantageous hex and assigned to combat. After that,
- ' this routine should NOT be called again!
- procedure weightAndMove(unitsUnableToMove, destHex)
- {
- var closestUnit, unitList, unitLookingAt, movementRemaining, movementAllowance,
- whichUnitToMove, unitHex, closestDist, distance, correctFacing,
- dir, retVal, potentialHexToAttack, canRotate, didAttack;
-
- write "Here in weightAndMove";
- ' First, create a list of units that can move. This involves checking
- ' the group list and NOT adding units in the "unitsUnableToMove" list
- ' and also excluding units that have already moved by this leader. This
- ' little routine also includes units that have friendlies in their
- ' frontal hexes
- closestUnit = -1; ' Illegal unit
- closestDist = 100; ' No one will be this far away.
- GroupListReset();
- unitList = GetList();
- unitLookingAt = GroupListGetNext();
- while (unitLookingAt != -1)
- {
- if (IsMemberOf(unitsUnableToMove, unitLookingAt) == FALSE)
- {
- movementRemaining = Query(5, unitLookingAt);
- movementAllowance = Query(4, unitLookingAt);
- if (movementRemaining == movementAllowance)
- {
- if (CanMove(unitLookingAt))
- {
- AddToList(unitList, unitLookingAt);
- unitHex = Query(REGION_NUM, unitLookingAt);
- distance = Distance(unitHex, destHex);
- if (distance < closestDist)
- {
- watch "This unit is closest to the destination: " unitLookingAt;
- watch "His distance to the destination is " distance;
- closestUnit = unitLookingAt;
- closestDist = distance;
- };
- };
- };
- };
- unitLookingAt = GroupListGetNext();
- };
-
- ' Check to see if there were any legal units to move. If not, then they
- ' are (probably) out of range. Move the leader...
- if (closestUnit == -1)
- {
- write "No more units to move... moving the leader.";
- closestUnit = GetLeader();
- retVal = MoveLeader();
- if (retVal == ALREADY_MOVED)
- {
- ' If I'm in here, that means that there were no more units within range
- ' to move before and after the leader movement.
- FinishLeader();
- return MOVE_SUCCESS;
- };
- if (retVal == 0)
- {
- ' If I'm in here, that means there were no legal moves for the leader
- ' and he couldn't move anyone else, so here it ends.
- FinishLeader();
- return MOVE_SUCCESS;
- };
- return retVal;
- };
-
- ' I have the closest unit now. See if he is facing the right direction.
- ' If he isn't, rotate him so he is facing the right direction.
- unitHex = Query(REGION_NUM, closestUnit);
- correctFacing = IsInGeneralDir(closestUnit, destHex);
- if (correctFacing == FALSE)
- {
- dir = Direction(unitHex, destHex);
- canRotate = CanRotate(closestUnit, dir);
- if (canRotate)
- {
- retVal = Rotate(closestUnit, dir);
- if (retVal == OW_MOVE)
- return MOVE_SUCCESS;
- };
- };
-
- ' Now, see if there are any units that are capable of being attacked...
- potentialHexToAttack = AnyoneToCombat(closestUnit);
- if (potentialHexToAttack != -1)
- {
- watch "Found a hex to attack: " potentialHexToAttack;
- retVal = MoveToward(potentialHexToAttack, closestUnit);
- if (retVal == 0)
- {
- return closestUnit;
- };
- retVal = TurnToAttack(closestUnit);
- if (retVal == OW_MOVE)
- {
- write "Turning to attack by there is OW involved. Quitting.";
- return MOVE_SUCCESS;
- };
- didAttack = AttackInZOC(closestUnit);
- if (NOT didAttack)
- ShootMissilesFar(closestUnit);
- return MOVE_SUCCESS;
- };
- retVal = MoveToward(destHex, closestUnit);
- ShootMissilesFar(closestUnit);
- if (retVal == 0)
- return closestUnit;
- return MOVE_SUCCESS;
- };
-
- procedure Beserk()
- {
- var inRangeList, unit, dir, correctFacing, unitHex, hex, canRotate, isRouting,
- noMoreUnits, retVal, f, newDir, didMove, didAttack, cohesionHits, tq,
- diff, rallied, recovered;
-
- write "Now in Beserk procedure";
- noMoreUnits = NoMoreUnitsToCommand();
- if (noMoreUnits)
- {
- CommandOtherUnits();
- return True;
- };
- inRangeList = GetList();
- GetUnitsInRange(inRangeList);
- ResetList(inRangeList);
- unit = GetNextInList(inRangeList);
- while (unit != -1)
- {
- isRouting = Query(ROUTING, unit);
- if (isRouting == False)
- {
- ' Check to see if he is close to routing. If he is, don't move him...
- watch "Now looking at unit #" unit;
- cohesionHits = Query(COHESION_HITS, unit);
- tq = Query(TQ, unit);
- diff = tq - cohesionHits;
- if (diff <= 2)
- watch "Not ordering him to move. He is close to routing..." diff
- else
- {
- hex = FindClosestEnemyToAttack(unit);
- watch "He can attack hex #" hex;
- if (hex != -1)
- {
- correctFacing = IsInGeneralDir(unit, hex);
- if (correctFacing == False)
- {
- write "The unit is NOT facing the correct direction";
- f = Query(FACING, unit);
- watch "His current facing: " f;
- unitHex = Query(REGION_NUM, unit);
- dir = Direction(unitHex, hex);
- watch "The direction to rotate to: " dir;
- canRotate = CanRotate(unit, dir);
- if (canRotate)
- {
- retVal = Rotate(unit, dir);
- if (retVal == OW_MOVE)
- return 1;
- };
- };
- didMove = MoveToward(hex, unit);
- newDir = TurnToAttack(unit);
- if (newDir == OW_MOVE)
- {
- write "Turning to attack (from Beserk) with OW. Quitting.";
- return 1;
- };
- watch "This unit about to (maybe) attack: " unit;
- 'if (newDir != NO_MOVE)
- didAttack = AttackInZOC(unit);
- if (didAttack)
- watch "This unit DID attack: " unit
- else
- watch "This unit did NOT attack: " unit;
- if ((NOT didAttack) AND didMove)
- ShootMissilesFar(unit);
- if (didMove OR didAttack)
- return 1;
- };
- };
- };
- unit = GetNextInList(inRangeList);
- };
-
- ' No one was or could be moved. Check to see if anyone needs to be rallied
- ' or restored with rally being given priority.
- rallied = Rally();
- if (rallied != -1)
- return 1;
- ' No one was rallied. Now try restoring cohesion hits
- recovered = RecoverUnits();
- if (recovered == -1)
- {
- retVal = MoveLeader();
- if (retVal == ALREADY_MOVED)
- FinishLeader();
- if (retVal == NO_MOVE)
- FinishLeader();
- };
- };
-
- procedure MoveLeader()
- {
- ' This procedure moves the leader. It will look first to see if it can move
- ' into the midst of his troops and have the majority of them be in command
- ' range.
- ' It will first look to see if the leader has moved before in this current
- ' orders phase. If he has, it will return the ALREADY_MOVED constant.
-
- var aveX, aveY, hex, unit, numUnits, totalX, totalY, retVal, leader;
- leader = GetLeader();
-
- retVal = CanBeOrdered(leader);
- if (retVal == False)
- return ALREADY_MOVED;
-
- ' First, get the average x and y hexes.
- totalX = 0;
- totalY = 0;
- numUnits = 0;
- GroupListReset();
- unit = GroupListGetNext();
- while (unit != -1)
- {
- hex = Query(REGION_NUM, unit);
- if (hex != -1)
- {
- totalX = totalX + hex/100;
- totalY = totalY + (hex - (hex/100)*100);
- numUnits = numUnits + 1;
- };
- unit = GroupListGetNext();
- };
- ' Rather than just returning, the leader should detect that it has no units
- ' that are surviving and go and try to help with another leader. But, for
- ' now (2/27/97), it is the way it is...
- if (numUnits==0)
- return 0;
- aveX = totalX/numUnits;
- aveY = totalY/numUnits;
- hex = aveX*100+aveY;
- retVal = MoveToward(hex, leader);
- return retVal;
- };
-
- procedure AttackWithSkirmishers();
-
- procedure GenericMove(hex)
- {
- var outOfRangeList, battlingList, unitRallied, unitRecovered, noMoreUnits,
- retVal, done;
-
- noMoreUnits = NoMoreUnitsToCommand();
- if (noMoreUnits)
- {
- CommandOtherUnits();
- return True;
- };
- watch "Here in GenericMove and moving to hex " hex;
- unitRallied = Rally();
- if (unitRallied != -1)
- {
- write "A unit was rallied from GenericMove";
- return 1;
- };
- unitRecovered = PartialRecover();
- if (unitRecovered)
- {
- write "A unit was recovered";
- return 1;
- };
- outOfRangeList = GetList();
- battlingList = GetList();
- GetUnitsOutOfRange(outOfRangeList);
- GetUnitsEngagedInBattle(battlingList);
-
- ' This loop will (should) insure that the A/I leaders will use all of their
- ' commands.
- done = False;
- while (NOT done)
- {
- retVal = weightAndMove(outOfRangeList, hex);
- if (retVal >= 0) ' This indicates that the unit # could NOT move...
- {
- AddToList(outOfRangeList, retVal);
- watch "Could not move this unit: " retVal;
- };
- if (retVal == MOVE_SUCCESS)
- {
- done = True;
- write "Successfully moved a unit in GenericMove";
- };
- };
- };
-
- procedure AttackWithSkirmishers()
- {
- ' Just end the leader for right now.
- write "Finishing the leader from inside of AttackWithSkirmishers";
- FinishLeader();
- return 1;
- };
-
- procedure AttackInZOC(movingUnit)
- {
- var zocHexes, unit, hex, enemy;
-
- zocHexes = GetList();
- AssignList(zocHexes, ZOC_HEXES, movingUnit);
- ResetList(zocHexes);
- hex = GetNextInList(zocHexes);
- while (hex!=-1)
- {
- unit = HexOccupancy(hex);
- if (unit != -1)
- {
- enemy = IsEnemy(movingUnit, unit);
- Attack(movingUnit, hex);
- ShootMissiles(movingUnit);
- return 1;
- };
- hex = GetNextInList(zocHexes);
- };
- return 0;
- };
-
- procedure ShootMissiles(shootingUnit)
- {
- var zocHexes, unit, hex, enemy, numMissiles;
- if (Query(MISSILE_CAPABLE, shootingUnit) == 0)
- {
- write "This unit CANNOT fire any missiles";
- return 0;
- };
- write "This unit CAN shoot missiles";
- numMissiles = Query(MISSILES_REMAINING, shootingUnit);
- if (numMissiles == 0)
- {
- write "This unit is out of missiles so he will not shoot";
- return 0;
- };
- ' Look in his ZOC and just see if he can shoot somebody...
- zocHexes = GetList();
- AssignList(zocHexes, ZOC_HEXES, shootingUnit);
- ResetList(zocHexes);
- hex = GetNextInList(zocHexes);
- while (hex!=-1)
- {
- unit = HexOccupancy(hex);
- if (unit != -1)
- {
- enemy = IsEnemy(shootingUnit, unit);
- if (enemy)
- {
- Fire(shootingUnit, hex);
- return 1;
- };
- };
- hex = GetNextInList(zocHexes);
- };
- return 0;
- };
-
- procedure ShootMissilesFar(unitToFire)
- {
- var hexToFireAt, canFire;
-
- canFire = Query(MISSILE_CAPABLE, unitToFire);
- if (canFire)
- {
- hexToFireAt = FindClosestEnemyToFire(unitToFire);
- if (hexToFireAt != -1)
- {
- Fire(unitToFire, hexToFireAt);
- return True;
- };
- };
- return False;
- };
-
- procedure Defend(hexToDefend)
- {
- ' First, get a list of units in range and a list of units out of range.
- var retVal, noMoreUnits, unit, legalUnits;
-
- write "Here in Defend";
- noMoreUnits = NoMoreUnitsToCommand();
- if (noMoreUnits)
- {
- CommandOtherUnits();
- return True;
- };
-
- retVal = Rally();
- if (retVal != -1)
- return 1;
-
- ' Look for units within 3 hits of routing... (excepting units that only have
- ' 3 TQ points).
- write "Couldn't find any units to rally!";
- retVal = PartialRecover();
- if (retVal)
- return 1;
-
- write "Couldn't find a unit to recover... now looking to rotate a unit...";
- retVal = TurnToDefend();
- if (retVal == True)
- return 1;
-
- write "Couldn't find a unit to turn to defend himself... now trying to fire missiles...";
- legalUnits = GetList();
- GetUnitsToCommand(legalUnits);
- unit = GetNextInList(legalUnits);
- while (unit != -1)
- {
- retVal = ShootMissilesFar(unit);
- if (retVal)
- return 1;
- unit = GetNextInList(legalUnits);
- };
-
- retVal = RecoverUnits();
- if (retVal == -1)
- FinishLeader();
- };
-
- procedure Rally()
- {
- ' This function attempts to rally anyone in the leader's range.
- ' The return value is 1 if someone was rallied and 0 if not.
-
- var outOfRangeUnits, numUnitsOutOfRange, routingUnitsInRange, unit, notInRange,
- routing, retVal;
-
- outOfRangeUnits = GetList();
- numUnitsOutOfRange = GetUnitsOutOfRange(outOfRangeUnits);
-
- retVal = 0;
- write "Here in Rally";
- ' Get a list of units that are in range and routing...
- routingUnitsInRange = GetList();
- GroupListReset();
- unit = GroupListGetNext();
- while (unit != -1)
- {
- notInRange = IsMemberOf(outOfRangeUnits, unit);
- routing = False;
- if (NOT notInRange)
- {
- watch "Looking at unit " unit;
- routing = Query(ROUTING, unit);
- if (routing)
- write "He IS routing"
- else
- write "He is NOT routing";
- };
- if (routing)
- {
- watch "Found a routing unit: " unit;
- retVal = AttemptToRally(unit);
- return retVal;
- };
- unit = GroupListGetNext();
- };
- write "Couldn't find any units to rally!";
- return -1;
- };
-
- procedure PartialRecover()
- {
- ' Find any unit that is within 3 cohesion hits of routing and recover him
-
- var unitsOutOfRange, numUnitsOutOfRange, notInRange, unit, tq, cohesionHit,
- difference, canOrder, isRouting;
-
- write "Here in PartialRecover";
- unitsOutOfRange = GetList();
- numUnitsOutOfRange = GetUnitsOutOfRange(unitsOutOfRange);
- GroupListReset();
- unit = GroupListGetNext();
- while (unit!=-1)
- {
- notInRange = IsMemberOf(unitsOutOfRange, unit);
- canOrder = CanBeOrdered(unit);
- if ((NOT notInRange) AND canOrder)
- {
- isRouting = Query(ROUTING, unit);
- if (NOT isRouting)
- {
- cohesionHit = Query(COHESION_HIT, unit);
- tq = Query(TROOP_Q, unit);
- difference = tq-cohesionHit;
- if ( (difference <= 3) AND (cohesionHit > 0))
- {
- Recover(unit);
- watch "Recovered unit: " unit;
- return 1;
- };
- };
- };
- unit = GroupListGetNext();
- };
- return 0;
- };
-
- procedure RecoverUnits()
- {
- ' Find all the units within range and recover he who has the most cohesion
- ' hits.
-
- var unitsOutOfRange, numUnitsOutOfRange, notInRange, unit, highUnit,
- highCH, cohesionHit, isRouting;
-
- write "Here in RecoverUnits";
- unitsOutOfRange = GetList();
- numUnitsOutOfRange = GetUnitsOutOfRange(unitsOutOfRange);
- highCH = 0;
- highUnit = -1;
- GroupListReset();
- unit = GroupListGetNext();
- while (unit != -1)
- {
- notInRange = IsMemberOf(unitsOutOfRange, unit);
- if (NOT notInRange)
- {
- watch "Looking at unit " unit;
- isRouting = Query(ROUTING, unit);
- if (NOT isRouting)
- {
- cohesionHit = Query(COHESION_HIT, unit);
- if (cohesionHit > highCH)
- {
- highCH = cohesionHit;
- highUnit = unit;
- watch "New high cohesion hit: " highCH;
- };
- };
- };
- unit = GroupListGetNext();
- };
- if (highUnit == -1)
- {
- write "There weren't any units that needed recovering...";
- return -1;
- };
- Recover(highUnit);
- return highUnit;
- };
-
- procedure RecoverAnyUnit()
- {
- ' Find all the units within range and recover he who has the most cohesion
- ' hits.
-
- var unit, highUnit, unitList, leader,
- highCH, cohesionHit, isRouting;
-
- write "Here in RecoverUnits";
- unitList = GetList();
- highCH = 0;
- highUnit = -1;
- leader = GetLeader();
- AssignList(unitList, IN_RANGE_UNITS, leader);
- unit = GetNextInList(unitList);
- while (unit != -1)
- {
- watch "Looking at unit " unit;
- isRouting = Query(ROUTING, unit);
- if (NOT isRouting)
- {
- cohesionHit = Query(COHESION_HIT, unit);
- if (cohesionHit > highCH)
- {
- highCH = cohesionHit;
- highUnit = unit;
- watch "New high cohesion hit: " highCH;
- };
- };
- unit = GetNextInList(unitList);
- };
- if (highUnit == -1)
- {
- write "There weren't any units that needed recovering...";
- return -1;
- };
- Recover(highUnit);
- return highUnit;
- };
-
- procedure EnemyWithinRangeOfGroup(radius)
- {
- var unit, inRange;
-
- GroupListReset();
- unit = GroupListGetNext();
- while (unit!=-1)
- {
- inRange = EnemyWithinRange(unit, radius);
- if (inRange)
- return True;
- unit = GroupListGetNext();
- };
- };
-
- procedure CommandOtherUnits()
- {
- var hexToMoveTo, unitToMoveTowards, initiative, leader, isRouting, inRange,
- moveValue, retVal;
-
- write "Here in CommandOtherUnits";
- hexToMoveTo = FindHurtingUnit();
- if (hexToMoveTo == -1)
- {
- write "Couldn't find any units to Rally or Restore. Now finishing the leader.";
- FinishLeader();
- return 0;
- };
- watch "Found this hex to move to: " hexToMoveTo;
- unitToMoveTowards = HexOccupancy(hexToMoveTo);
- leader = GetLeader();
- initiative = Query(INITIATIVE, leader);
- inRange = InCommandRange(unitToMoveTowards);
- if (NOT inRange)
- {
- write "The unit was NOT in range so the leader is being moved";
- moveValue = MoveToward(hexToMoveTo, leader);
- if (moveValue == 0)
- {
- retVal = RecoverAnyUnit();
- if (retVal == -1)
- FinishLeader();
- };
- return 1;
- };
- if (initiative>=4)
- {
- isRouting = Query(ROUTING, unitToMoveTowards);
- if (isRouting)
- {
- AttemptToRally(unitToMoveTowards);
- return 1;
- };
- };
- Recover(unitToMoveTowards);
- };
-
- procedure NoMoreUnitsToCommand()
- {
- var groupListSize;
-
- groupListSize = GroupListGetSize();
- if (groupListSize == 0)
- return True;
- return False;
- };
-
- procedure TurnToDefend()
- {
- var unitsToCommand, unit, enemyHex, hex, d, shortDistance, highUnit,
- high, highEnemyHex, highHex, retVal;
-
- write "Here in TurnToDefend";
- unitsToCommand = GetList();
- GetUnitsToCommand(unitsToCommand);
- shortDistance = 999;
- highUnit = -1;
- ResetList(unitsToCommand);
- unit = GetNextInList(unitsToCommand);
- while (unit != -1)
- {
- hex = Query(REGION_NUM, unit);
- enemyHex = FindClosestEnemyToAttack(unit);
- retVal = IsInGeneralDir(unit, enemyHex);
- if (retVal == False)
- {
- d = Distance(enemyHex, hex);
- if (shortDistance > d)
- {
- shortDistance = d;
- highUnit = unit;
- highHex = hex;
- highEnemyHex = enemyHex;
- };
- };
- unit = GetNextInList(unitsToCommand);
- };
- if (highUnit != -1)
- {
- d = Direction(highHex, highEnemyHex);
- retVal = CanRotate(highUnit, d);
- if (retVal == False)
- {
- watch "Unit cannot rotate for some reason: " highUnit;
- return False;
- };
- Rotate(highUnit, d);
- watch "Rotated this unit: " highUnit;
- ' Now check to see if he can fire any missiles...
- return True;
- };
- return False;
- };
-
- procedure GetUnitsToCommand(legalList)
- {
- var unitsOutOfRange, unit, illegalUnits, retVal;
-
- GroupListReset();
- unit = GroupListGetNext();
- while (unit != -1)
- {
- AddToList(legalList, unit);
- unit = GroupListGetNext();
- };
-
- ' First, remove all units that are out of range...
- unitsOutOfRange = GetList();
- GetUnitsOutOfRange(unitsOutOfRange);
- unit = GetNextInList(unitsOutOfRange);
- while (unit != -1)
- {
- RemoveFromList(legalList, unit);
- unit = GetNextInList(unitsOutOfRange);
- };
-
- ' Now, remove units that cannot be commanded...
- ResetList(legalList);
- illegalUnits = GetList();
- unit = GetNextInList(legalList);
- while (unit != -1)
- {
- retVal = CanBeOrdered(unit);
- if (retVal == False)
- AddToList(illegalUnits, unit);
- unit = GetNextInList(legalList);
- };
- ResetList(illegalUnits);
- unit = GetNextInList(illegalUnits);
- while (unit != -1)
- {
- RemoveFromList(legalList, unit);
- unit = GetNextInList(illegalUnits);
- };
- return 1;
- };
-